home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / iinit.c < prev    next >
C/C++ Source or Header  |  1997-04-11  |  14KB  |  463 lines

  1. /* Copyright (C) 1989, 1995, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* iinit.c */
  20. /* Initialize internally known objects for Ghostscript interpreter */
  21. #include "string_.h"
  22. #include "ghost.h"
  23. #include "gscdefs.h"
  24. #include "gsexit.h"
  25. #include "gsstruct.h"
  26. #define INCLUDE_ERROR_NAMES        /* see errors.h */
  27. #include "errors.h"
  28. #include "ialloc.h"
  29. #include "idict.h"
  30. #include "dstack.h"
  31. #include "ilevel.h"
  32. #include "iname.h"
  33. #include "interp.h"
  34. #include "ipacked.h"
  35. #include "iparray.h"
  36. #include "iutil.h"
  37. #include "ivmspace.h"
  38. #include "opdef.h"
  39. #include "store.h"
  40.  
  41. /* Implementation parameters. */
  42. /* The size of systemdict can be set in the makefile. */
  43. /* We want the sizes to be prime numbers large enough to cover */
  44. /* all the operators, plus everything in the init files, */
  45. /* even if all the optional features are selected. */
  46. #ifndef SYSTEMDICT_SIZE
  47. #  define SYSTEMDICT_SIZE 563
  48. #endif
  49. #ifndef SYSTEMDICT_LEVEL2_SIZE
  50. #  define SYSTEMDICT_LEVEL2_SIZE 809
  51. #endif
  52. /* The size of level2dict, if applicable, can be set in the makefile. */
  53. #ifndef LEVEL2DICT_SIZE
  54. #  define LEVEL2DICT_SIZE 191
  55. #endif
  56. /* Ditto the size of filterdict. */
  57. #ifndef FILTERDICT_SIZE
  58. #  define FILTERDICT_SIZE 43
  59. #endif
  60. /* Ditto the size of internaldict. */
  61. #ifndef INTERNALDICT_SIZE
  62. #  define INTERNALDICT_SIZE 3
  63. #endif
  64. /* Define an arbitrary size for the operator procedure tables. */
  65. #ifndef OP_ARRAY_TABLE_SIZE
  66. #  define OP_ARRAY_TABLE_SIZE 120
  67. #endif
  68. #ifndef OP_ARRAY_TABLE_GLOBAL_SIZE
  69. #  define OP_ARRAY_TABLE_GLOBAL_SIZE OP_ARRAY_TABLE_SIZE
  70. #endif
  71. #ifndef OP_ARRAY_TABLE_LOCAL_SIZE
  72. #  define OP_ARRAY_TABLE_LOCAL_SIZE (OP_ARRAY_TABLE_SIZE / 2)
  73. #endif
  74. #define OP_ARRAY_TABLE_TOTAL_SIZE\
  75.   (OP_ARRAY_TABLE_GLOBAL_SIZE + OP_ARRAY_TABLE_LOCAL_SIZE)
  76.  
  77. /* The operator tables */
  78. extern op_def_ptr (*(op_defs_all[]))(P0());    /* in iconfig.c */
  79. /* Because of a bug in Sun's SC1.0 compiler, */
  80. /* we have to spell out the typedef for op_def_ptr here: */
  81. const op_def **op_def_table;
  82. uint op_def_count;
  83. op_array_table op_array_table_global, op_array_table_local;    /* definitions of `operator' procedures */
  84. uint op_array_count;
  85. /* GC roots for the same */
  86. private gs_gc_root_t
  87.   op_def_root, op_array_root_global, op_array_root_local,
  88.   op_array_nx_root_global, op_array_nx_root_local;
  89.  
  90. /* Enter a name and value into a dictionary. */
  91. void
  92. initial_enter_name_in(const char *nstr, const ref *pref, ref *pdict)
  93. {    int code = dict_put_string(pdict, nstr, pref);
  94.     if ( code < 0 )
  95.       {    lprintf4("initial_enter failed (%d), entering /%s in -dict:%u/%u-\n",
  96.              code, nstr, dict_length(pdict), dict_maxlength(pdict));
  97.         gs_exit(1);
  98.       }
  99. }
  100. void
  101. initial_enter_name(const char *nstr, const ref *pref)
  102. {    initial_enter_name_in(nstr, pref, systemdict);
  103. }
  104.  
  105. /* Remove a name from systemdict. */
  106. void
  107. initial_remove_name(const char *nstr)
  108. {    ref nref;
  109.     if ( name_ref((const byte *)nstr, strlen(nstr), &nref, -1) >= 0 )
  110.       dict_undef(systemdict, &nref);
  111. }
  112.  
  113. /* Create a name.  Fatal error if it fails. */
  114. private void
  115. name_enter(const char *str, ref *pref)
  116. {    if ( name_enter_string(str, pref) != 0 )
  117.       {    lprintf1("name_enter failed - %s\n", str);
  118.         gs_exit(1);
  119.       }
  120. }
  121.  
  122. /* Define the names and sizes of the initial dictionaries. */
  123. /* The names are used to create references in systemdict. */
  124. const struct { const char *name; uint size; bool local; }
  125.   initial_dictionaries[] = {
  126. #ifdef INITIAL_DICTIONARIES
  127.     INITIAL_DICTIONARIES
  128. #else
  129.     /* systemdict is created and named automagically */
  130.     { "level2dict", LEVEL2DICT_SIZE, false },
  131.     { "globaldict", 0, false },
  132.     { "userdict", 0, true },
  133.     { "filterdict", FILTERDICT_SIZE, false },
  134.     { "internaldict", INTERNALDICT_SIZE, true }
  135. #endif
  136. };
  137. /* systemdict and globaldict are magically inserted at the bottom */
  138. const char *initial_dstack[] = {
  139. #ifdef INITIAL_DSTACK
  140.     INITIAL_DSTACK
  141. #else
  142.     "userdict"
  143. #endif
  144. };
  145. #define MIN_DSTACK_SIZE (countof(initial_dstack) + 1) /* +1 for systemdict */
  146.  
  147.  
  148. /* Detect whether we have any Level 2 operators. */
  149. /* We export this for gs_init1 in gsmain.c. */
  150. /* This is very slow, but we only call it a couple of times. */
  151. bool
  152. gs_have_level2(void)
  153. {    op_def_ptr (**tptr)(P0());
  154.  
  155.     for ( tptr = op_defs_all; *tptr != 0; tptr++ )
  156.      {    const op_def *def;
  157.         for ( def = (*tptr)(); def->oname != 0; def++ )
  158.           if ( op_def_is_begin_dict(def) &&
  159.                !strcmp(def->oname, "level2dict")
  160.              )
  161.             return true;
  162.      }
  163.     return false;
  164. }
  165.  
  166. /* Create an initial dictionary if necessary. */
  167. private ref *
  168. make_initial_dict(const char *iname, ref idicts[])
  169. {    int i;
  170.  
  171.     /* systemdict was created specially. */
  172.     if ( !strcmp(iname, "systemdict") )
  173.       return &ref_systemdict;
  174.     for ( i = 0; i < countof(initial_dictionaries); i++ )
  175.       {    const char *dname = initial_dictionaries[i].name;
  176.         const int dsize = initial_dictionaries[i].size;
  177.  
  178.         if ( !strcmp(iname, dname) )
  179.           {    ref *dref = &idicts[i];
  180.             if ( r_has_type(dref, t_null) )
  181.               {    int code;
  182.                 /* Perhaps dict_create should take */
  183.                 /* the allocator as an argument.... */
  184.                 uint space = ialloc_space(idmemory);
  185.                 ialloc_set_space(idmemory,
  186.                     (initial_dictionaries[i].local ?
  187.                      avm_local : avm_global));
  188.                 code = dict_create(dsize, dref);
  189.                 ialloc_set_space(idmemory, space);
  190.                 if ( code < 0 )
  191.                   return 0;    /* disaster */
  192.               }
  193.             return dref;
  194.           }
  195.       }
  196.  
  197.     /*
  198.      * Name mentioned in some op_def, but not in initial_dictionaries.
  199.      * Punt.
  200.      */
  201.     return 0;
  202. }
  203.  
  204. /* Initialize objects other than operators.  In particular, */
  205. /* initialize the dictionaries that hold operator definitions. */
  206. void
  207. obj_init(void)
  208. {    uint space = ialloc_space(idmemory);
  209.     bool level2 = gs_have_level2();
  210.  
  211.     /* Initialize the language level. */
  212.     make_int(&ref_language_level, 1);
  213.  
  214.     /* Initialize the interpreter. */
  215.     gs_interp_init();
  216.  
  217.     {
  218. #define icount countof(initial_dictionaries)
  219.       ref idicts[icount];
  220.       int i;
  221.       op_def_ptr (**tptr)(P0());
  222.  
  223.       min_dstack_size = MIN_DSTACK_SIZE;
  224.  
  225.       refset_null(idicts, icount);
  226.  
  227.       /* Create systemdict. */
  228.       ialloc_set_space(idmemory, avm_global);
  229.       if ( level2 )
  230.         {    dsp += 2;
  231.         dict_create(SYSTEMDICT_LEVEL2_SIZE, dsp);
  232.         /* For the moment, let globaldict be an alias */
  233.         /* for systemdict. */
  234.         dsp[-1] = *dsp;
  235.         min_dstack_size++;
  236.         }
  237.       else
  238.         {    ++dsp;
  239.         dict_create(SYSTEMDICT_SIZE, dsp);
  240.         }
  241.       ref_systemdict = *dsp;
  242.       ialloc_set_space(idmemory, space);
  243.  
  244.       /* Create dictionaries which are to be homes for operators. */
  245.       for ( tptr = op_defs_all; *tptr != 0; tptr++ )
  246.         {    const op_def *def;
  247.         for ( def = (*tptr)(); def->oname != 0; def++ )
  248.           if ( op_def_is_begin_dict(def) )
  249.             make_initial_dict(def->oname, idicts);
  250.         }
  251.  
  252.       /* Set up the initial dstack. */
  253.       for ( i = 0; i < countof(initial_dstack); i++ )
  254.         {    const char *dname = initial_dstack[i];
  255.         ++dsp;
  256.         ref_assign(dsp, make_initial_dict(dname, idicts));
  257.         }
  258.  
  259.       /* Enter names of referenced initial dictionaries into systemdict. */
  260.       initial_enter_name("systemdict", &ref_systemdict);
  261.       for (i = 0; i < icount; i++)
  262.         {    ref *idict = &idicts[i];
  263.         if ( !r_has_type(idict, t_null) )
  264.           {    /*
  265.              * Note that we enter the dictionary in systemdict
  266.              * even if it is in local VM.  There is a special
  267.              * provision in the garbage collector for this:
  268.              * see ivmspace.h for more information.
  269.              * In order to do this, we must temporarily
  270.              * identify systemdict as local, so that the
  271.              * store check in dict_put won't fail.
  272.              */
  273.             uint save_space = r_space(systemdict);
  274.             r_set_space(systemdict, avm_local);
  275.             initial_enter_name(initial_dictionaries[i].name,
  276.                        idict);
  277.             r_set_space(systemdict, save_space);
  278.           }
  279.         }
  280. #undef icount
  281.     }
  282.  
  283.     gs_interp_reset();
  284.  
  285.     {    ref vtemp;
  286.         make_null(&vtemp);
  287.         initial_enter_name("null", &vtemp);
  288.         make_true(&vtemp);
  289.         initial_enter_name("true", &vtemp);
  290.         make_false(&vtemp);
  291.         initial_enter_name("false", &vtemp);
  292.     }
  293.  
  294.     /* Create the error name table */
  295.     {    int n = countof(gs_error_names) - 1;
  296.         int i;
  297.         ref era;
  298.         ialloc_ref_array(&era, a_readonly, n, "ErrorNames");
  299.         for ( i = 0; i < n; i++ )
  300.           name_enter((const char *)gs_error_names[i],
  301.                  era.value.refs + i);
  302.         initial_enter_name("ErrorNames", &era);
  303.     }
  304. }
  305.  
  306. /* Run the initialization procedures of the individual operator files. */
  307. void
  308. zop_init(void)
  309. {    op_def_ptr (**tptr)(P0());
  310.     /* Because of a bug in Sun's SC1.0 compiler, */
  311.     /* we have to spell out the typedef for op_def_ptr here: */
  312.     const op_def *def;
  313.     for ( tptr = op_defs_all; *tptr != 0; tptr++ )
  314.       {    for ( def = (*tptr)(); def->oname != 0; def++ )
  315.           DO_NOTHING;
  316.         if ( def->proc != 0 )
  317.           ((void (*)(P0()))(def->proc))();
  318.       }
  319.  
  320.     /* Initialize the predefined names other than operators. */
  321.     /* Do this here in case op_init changed any of them. */
  322.     {    ref vtemp;
  323.         make_const_string(&vtemp, a_readonly | avm_foreign,
  324.                   strlen(gs_copyright),
  325.                   (const byte *)gs_copyright);
  326.         initial_enter_name("copyright", &vtemp);
  327.         make_const_string(&vtemp, a_readonly | avm_foreign,
  328.                   strlen(gs_product),
  329.                   (const byte *)gs_product);
  330.         initial_enter_name("product", &vtemp);
  331.         make_int(&vtemp, gs_revision);
  332.         initial_enter_name("revision", &vtemp);
  333.         make_int(&vtemp, gs_revisiondate);
  334.         initial_enter_name("revisiondate", &vtemp);
  335.     }
  336. }
  337.  
  338. /* Create an op_array table. */
  339. private int
  340. alloc_op_array_table(uint size, uint space, op_array_table *opt)
  341. {    uint save_space = ialloc_space(idmemory);
  342.     int code;
  343.     ialloc_set_space(idmemory, space);
  344.     code = ialloc_ref_array(&opt->table, a_readonly, size,
  345.                 "op_array table");
  346.     ialloc_set_space(idmemory, save_space);
  347.     if ( code < 0 )
  348.       return code;
  349.     refset_null(opt->table.value.refs, size);
  350.     opt->nx_table =
  351.       (ushort *)ialloc_byte_array(size, sizeof(ushort),
  352.                       "op_array nx_table");
  353.     if ( opt->nx_table == 0 )
  354.       return_error(e_VMerror);
  355.     opt->count = 0;
  356.     opt->root_p = &opt->table;
  357.     opt->attrs = space | a_executable;
  358.     return 0;
  359. }
  360.  
  361. /* Initialize the operator table. */
  362. void
  363. op_init(void)
  364. {    int count = 1;
  365.     op_def_ptr (**tptr)(P0());
  366.     /* Because of a bug in Sun's SC1.0 compiler, */
  367.     /* we have to spell out the typedef for op_def_ptr here: */
  368.     const op_def *def;
  369.     const char _ds *nstr;
  370.  
  371.     /* Do a first pass just to count the operators. */
  372.  
  373.     for ( tptr = op_defs_all; *tptr != 0; tptr ++ )
  374.     {    for ( def = (*tptr)(); def->oname != 0; def++ )
  375.           if ( !op_def_is_begin_dict(def) )
  376.             count++;
  377.     }
  378.     
  379.     /* Do a second pass to construct the operator table, */
  380.     /* and enter the operators into the appropriate dictionary. */
  381.  
  382.     /* Because of a bug in Sun's SC1.0 compiler, */
  383.     /* we have to spell out the typedef for op_def_ptr here: */
  384.     op_def_table =
  385.       (const op_def **)ialloc_byte_array(count, sizeof(op_def_ptr),
  386.                          "op_init(op_def_table)");
  387.     op_def_count = count;
  388.     for (count = 0; count <= gs_interp_num_special_ops; count++)
  389.       op_def_table[count] = 0;
  390.     count = gs_interp_num_special_ops + 1; /* leave space for magic entries */
  391.     for ( tptr = op_defs_all; *tptr != 0; tptr ++ )
  392.       {    ref *pdict = systemdict;
  393.         for ( def = (*tptr)(); (nstr = def->oname) != 0; def++ )
  394.           if ( op_def_is_begin_dict(def) )
  395.             {    ref nref;
  396.             int code = name_ref((const byte *)nstr, strlen(nstr),
  397.                         &nref, -1);
  398.  
  399.             if ( code != 0 )
  400.               gs_abort();
  401.             if ( !dict_find(systemdict, &nref, &pdict) )
  402.               gs_abort();
  403.             if ( !r_has_type(pdict, t_dictionary) )
  404.               gs_abort();
  405.             }
  406.           else
  407.             {    ref oper;
  408.             uint opidx;
  409.  
  410.             gs_interp_make_oper(&oper, def->proc, count);
  411.             opidx = r_size(&oper);
  412.             /* The first character of the name is a digit */
  413.             /* giving the minimum acceptable number of operands. */
  414.             /* Check to make sure it's within bounds. */
  415.             if ( *nstr - '0' > gs_interp_max_op_num_args )
  416.               gs_abort();
  417.             nstr++;
  418.             /* Don't enter internal operators into */
  419.             /* the dictionary. */
  420.             if ( *nstr != '%' )
  421.               initial_enter_name_in(nstr, &oper, pdict);
  422.             op_def_table[opidx] = def;
  423.             if ( opidx == count )
  424.               count++;
  425.             }
  426.       }
  427.     /* All of the built-ins had better be defined somewhere, */
  428.     /* or things like op_find_index will choke. */
  429.     for ( count = 1; count <= gs_interp_num_special_ops; count++ )
  430.       if ( op_def_table[count] == 0 )
  431.         gs_abort();
  432.     gs_register_struct_root(imemory, &op_def_root,
  433.                 (void **)&op_def_table, "op_def_table");
  434.  
  435.     /* Allocate the tables for `operator' procedures. */
  436.     /* Make one of them local so we can have local operators. */
  437.  
  438.     if ( alloc_op_array_table(OP_ARRAY_TABLE_GLOBAL_SIZE,
  439.                   avm_global, &op_array_table_global) < 0 )
  440.       gs_abort();
  441.     op_array_table_global.base_index = op_def_count;
  442.     gs_register_ref_root(imemory, &op_array_root_global,
  443.                  (void **)&op_array_table_global.root_p,
  444.                  "op_array_table(global)");
  445.     gs_register_struct_root(imemory, &op_array_nx_root_global,
  446.                 (void **)&op_array_table_global.nx_table,
  447.                 "op_array nx_table(global)");
  448.  
  449.     if ( alloc_op_array_table(OP_ARRAY_TABLE_LOCAL_SIZE,
  450.                   avm_local, &op_array_table_local) < 0 )
  451.       gs_abort();
  452.     op_array_table_local.base_index =
  453.       op_array_table_global.base_index +
  454.         r_size(&op_array_table_global.table);
  455.     gs_register_ref_root(imemory, &op_array_root_local,
  456.                  (void **)&op_array_table_local.root_p,
  457.                  "op_array_table(local)");
  458.     gs_register_struct_root(imemory, &op_array_nx_root_local,
  459.                 (void **)&op_array_table_local.nx_table,
  460.                 "op_array nx_table(local)");
  461.  
  462. }
  463.